# D16 Processor Reference Manual

Michael Nolan

Aug 16, 2016

## Contents

| 1 | The               | Processor                  | 1      |
|---|-------------------|----------------------------|--------|
|   | 1.1               | General Purpose Registers  | 1      |
|   | 1.2               | Flags                      | 1      |
| 2 | Inst              | ruction Set                | 1      |
|   | 2.1               | ADD                        | 1      |
|   | 2.2               | SUB                        | 1      |
|   | 2.3               | PUSH                       | 2      |
|   | _                 | POP                        | 2      |
|   | 2.5               | MOV                        | 2      |
|   |                   | 2.5.1 general MOV encoding | 2      |
|   |                   | 2.5.2 special byte MOV     | 2      |
|   | 2.6               | AND                        | 2      |
|   | $\frac{2.0}{2.7}$ | OR                         | 3      |
|   | 2.8               | XOR                        | 3      |
|   | 2.9               | NOT                        | 3      |
|   |                   | NEG                        | 3      |
|   |                   | LD                         | 3      |
|   |                   | ST                         | 4      |
|   |                   | CMP                        | 4      |
|   |                   | JMP                        | 4      |
|   |                   | CALL                       | 4      |
|   |                   | RET                        | 4      |
|   |                   |                            | 5      |
|   |                   | SHL                        | 5      |
|   |                   |                            | 5<br>5 |
|   |                   | ROL                        | 5<br>5 |
|   |                   | RCL                        | _      |
|   |                   | SET                        | 5      |
|   |                   | TEST                       | 5      |
|   | 2.23              | PUSHLR                     | 6      |
| 3 | Con               | dition Codes               | 6      |
| 1 | Don:              | ipherals                   | 7      |
| 4 |                   | UART                       | 7      |
|   | 4.1               | 4.1.1 IO_UART_DATA         | 7      |
|   |                   | 4.1.1 IO_UART_DATA         | 7      |
|   |                   | 4.1.3 IO_UART_BAUD         | 7      |
|   |                   | 4.1.0 IU_UAI(I_DAUD        | - 1    |

## 1 The Processor

The D16 Processor is a very simple, RISC like 16 bit processor with variable length instructions. It has 8 general purpose registers, 32 special purpose registers, and support for up to 64K of memory.

## 1.1 General Purpose Registers

The D16 processor defines 8 general purpose registers, called r0 - r7. 2 of these, although they behave the same as the other registers, have special meaning to the processor and in the ABI, and they are as follows:

**r6:** This is generally used as the pointer to the start of a stack frame, but has no special meaning to the processor

r7: This is the stack pointer, and is manipulated via the stack instructions (push and pop)

## 1.2 Flags

The processor also contains several flags in Special Register 0.

| Zero     | set if the result of the last computation is 0             |
|----------|------------------------------------------------------------|
| Sign     | set if the result is negative (bit 15 is set)              |
| Carry    | set if there was a carry or borrow in the past computation |
| oVerflow | set if there was a signed overflow in the last computation |

## 2 Instruction Set

Most instructions come in 2 formats, register and immediate. The immediate versions of an instruction will have bit 7 set in the opcode field and the 16 bit immediate in the word following the instruction. In the subsequent definitions, op2 will refer to the immediate value if the instruction has an immediate, otherwise it refers to rS.

## 2.1 ADD

|   | Immediate            | opcode | Reserved | source | dest             |
|---|----------------------|--------|----------|--------|------------------|
| ĺ | $\operatorname{Imm}$ | 000001 | 00       | rS     | $^{\mathrm{rD}}$ |

ADD rD, <rS or immediate>

$$rD = rD + op2$$
  
Updates flags

## 2.2 SUB

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 000010 | 00       | rS     | $^{\mathrm{rD}}$ |

SUB rD, <rS or immediate>

$$rD = rD - op2$$
  
Updates flags

## 2.3 PUSH

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 000011 | 00       | 000    | $^{\mathrm{rD}}$ |

PUSH <rD or immediate>

 $\mathrm{r7}=\mathrm{r7}$  - 2

memory[r7] = rD

This instruction does not update the flags

## 2.4 POP

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 000100 | 00       | 000    | $^{\mathrm{rD}}$ |

POP <rD>

rD = memory[r7]

r7 = r7 + 2

Does not update flags

## 2.5 MOV

Mov has 2 different encodings depending whether the immediate (if any) fits into 1 byte. Neither encoding updates the flags.

## 2.5.1 general MOV encoding

| Immediate   | opcode | Reserved | source | dest |
|-------------|--------|----------|--------|------|
| $_{ m Imm}$ | 001101 | 00       | rS     | rD   |

MOV rD, <rS or immediate>

rD = op2

This encoding is used for register to register MOVs or when the immediate value will not fit in one byte.

## 2.5.2 special byte MOV

| Reserved | opcode      | data           |
|----------|-------------|----------------|
| 0        | 000101 + rD | byte immediate |

MOV rD, <byte immediate>

rD = immediate

This encoding is only used when the immediate will fit in 1 byte

## 2.6 AND

| Immediate   | opcode | Reserved | source | dest             |
|-------------|--------|----------|--------|------------------|
| $_{ m Imm}$ | 001110 | 00       | rS     | $^{\mathrm{rD}}$ |

AND rD, <rS or immediate>

## rD = rD AND op2

This instruction updates the flags, and will reset the overflow and carry flags.

## 2.7 OR

| Immediate | opcode | Reserved | source | dest |
|-----------|--------|----------|--------|------|
| Imm       | 001111 | 00       | rS     | rD   |

OR rD, <rS or immediate>

## rD = rD OR op2

This instruction updates the flags, and will reset the overflow and carry flags.

## 2.8 XOR

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 010000 | 00       | rS     | $^{\mathrm{rD}}$ |

XOR rD, <rS or immediate>

## rD = rD XOR op2

This instruction updates the flags, and will reset the overflow and carry flags.

## 2.9 NOT

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 010001 | 00       | 000    | $^{\mathrm{rD}}$ |

NOT <rD>

rD = !rD (bitwise NOT)

This instruction updates the flags, and will reset the overflow and carry flags.

## 2.10 NEG

| Immediate            | opcode | Reserved | source | dest             |
|----------------------|--------|----------|--------|------------------|
| $\operatorname{Imm}$ | 010010 | 00       | 000    | $^{\mathrm{rD}}$ |

NEG <rD>

rD = 0-rD (signed negation)

This instruction updates the flags.

## 2.11 LD

| ĺ | Immediate | opcode | byte | displacement | address | data |
|---|-----------|--------|------|--------------|---------|------|
| ĺ | imm       | 010011 | byte | disp         | rS      | rD   |

 $LD\{.b\}$  rD, [rS]

LD{.b} rD, [immediate]

LD{.b} rD, [rS+immediate]

The load instruction loads a word (or byte) of data from the address specified in the brackets into register rD. The byte flag in the instruction encoding is set when the ".b" suffix is present and indicates a byte load. The displacement flag is set when the third instruction form is used and indicates that rS must be added to the displacement when generating the address. The displacement flag should only be set if the immediate flag is also set. If the displacement flag is set and the immediate flag is not, the behavior is undefined. Important: All word accesses must be word aligned. Failure to ensure this will result in undefined behavior.

#### 2.12 ST

| Immediate   | opcode | byte | displacement | address | data |
|-------------|--------|------|--------------|---------|------|
| $_{ m imm}$ | 010100 | byte | disp         | rS      | rD   |

ST{.b} [rS], rD

ST{.b} [immediate], rD

ST{.b} [rS+immediate], rD

The store instruction stores the contents of rD into the address specified inside the brackets. The byte flag in the instruction is set when the ".b" suffix is present, and so the processor will only store the least significant 8 bits into the specified address. Similarly to the LD instruction, the disp flag indicates that the processor should add rS and the following immediate value before using the result as the address. Important: All word accesses must be word aligned. Failure to ensure this will result in undefined behavior.

#### 2.13 CMP

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 010101 | 00       | rS     | $^{\mathrm{rD}}$ |

CMP rD, <rS or immediate>

The instruction sets the flags exactly like the SUB instruction, however it does not store the result back to rD.

#### 2.14 JMP

| Immediate            | opcode | Reserved | condition code | dest |
|----------------------|--------|----------|----------------|------|
| $\operatorname{Imm}$ | 010110 | 0        | cc             | rD   |

JMP.CC <rD or immediate>

If the condition code evaluates to True, JMP sets the instruction pointer to the address specified. **Important: This address must be word aligned** 

#### 2.15 CALL

| Immediate   | opcode | Reserved | condition code | dest             |
|-------------|--------|----------|----------------|------------------|
| $_{ m Imm}$ | 010111 | 0        | cc             | $^{\mathrm{rD}}$ |

CALL.CC <rD or immediate>

If the condition evaluates to true, CALL saves the instruction pointer in the link register (LR), before setting it to the address specified. **Important: This address must be word aligned** 

## 2.16 RET

| Reserved | Opcode | Reserved |
|----------|--------|----------|
| 0        | 011000 | 00000000 |

RET

Sets the instruction pointer to the link register(LR).

## 2.17 SHL

| Immediate   | opcode | Reserved | source | dest             |
|-------------|--------|----------|--------|------------------|
| $_{ m Imm}$ | 011001 | 00       | rS     | $^{\mathrm{rD}}$ |

SHL rD, <rS or immediate>

Logical left shift rD by op2 and store the result in rD. Update flags and clear V.

## 2.18 SHR

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 011010 | 00       | rS     | $^{\mathrm{rD}}$ |

SHR rD, <rS or immediate>

Logical right shift rD by op2 and store the result in rD. Update flags and clear V.

#### 2.19 ROL

| Immediate | opcode | Reserved | source | dest             |
|-----------|--------|----------|--------|------------------|
| Imm       | 011011 | 00       | rS     | $^{\mathrm{rD}}$ |

ROL rD, <rS or immediate>

Rotates the bits in rD left by the specified number in op2. Updates flags and clears carry and overflow.

## 2.20 RCL

|   | Immediate | opcode | Reserved | source | dest             |
|---|-----------|--------|----------|--------|------------------|
| ĺ | Imm       | 011100 | 00       | rS     | $^{\mathrm{rD}}$ |

RCL rD, <rS or immediate>

Rotates the bits in rD, plus the carry bit left by the value in op2. Updates flags and clears overflow.

## 2.21 SET

| Immediate            | opcode | Reserved | condition code | dest |
|----------------------|--------|----------|----------------|------|
| $\operatorname{Imm}$ | 100001 | 0        | cc             | rD   |

SET.CC <rDI>

f the condition code evaluates to true, sets rD to 1 otherwise sets rD to 0.

## 2.22 TEST

| Immediate   | opcode | Reserved | source | dest             |
|-------------|--------|----------|--------|------------------|
| $_{ m Imm}$ | 100010 | 00       | rS     | $^{\mathrm{rD}}$ |

TEST rD, <rS or immediate>

Sets the flags upon the result of the bitwise and of rD and op2. Does not modify the value of rD.

## 2.23 PUSHLR

| Reserved | Opcode | Reserved |
|----------|--------|----------|
| 0        | 100011 | 00000000 |

PUSHLR

Pushes the link register onto the stack. This instruction should be used at the beginning of a subroutine if the subroutine executes the CALL instruction anywhere in its body. To return after executing this instruction, execute

POP r1

JMP.AL r1

Note: r1 can be replaced by any GP register

## 3 Condition Codes

In the JMP variety of instructions (JMP, CALL, SET), the condition code field specifies that the instruction should only execute if the expression corresponding to the condition code evaluates to true. The conditions are as follows:

| Mnemonic | Encoding | Expression                   | Meaning                         |
|----------|----------|------------------------------|---------------------------------|
| NV       | 0000     | False                        | Never execute                   |
| EQ       | 0001     | Z set                        | Equal                           |
| NE       | 0010     | Z clear                      | Not equal                       |
| OS       | 0011     | V set                        | Overflow                        |
| OC       | 0100     | V clear                      | No overflow                     |
| HI       | 0101     | C set and Z clear            | Unsigned greater than           |
| LS       | 0110     | C clear and Z set            | Unsigned less than or equal to  |
| P        | 0111     | S clear                      | Positive                        |
| N        | 1000     | S set                        | Negative                        |
| CS       | 1001     | C set                        | Carry set                       |
| CC       | 1010     | C clear                      | Carry clear                     |
| GE       | 1011     | S = V                        | Signed greater than or equal to |
| G        | 1100     | S = V and $Z$ clear          | Signed greater than             |
| LE       | 1101     | $Z \text{ set or } S \neq V$ | Signed less than or equal to    |
| L        | 1110     | $S \neq V$                   | Signed less than                |
| AL       | 1111     | True                         | Always execute                  |

## 4 Peripherals

The D16 processor uses memory mapping to access its peripherals. Currently, the memory mapped region starts at address 0xFF00 and ends at address 0xFFFF. However in subsequent revisions, this may be expanded to the range 0xFC00-0xFFFF, so the programmer is encouraged not to make use of the top 1K of the address space. All peripheral registers will specify their size, either 16 bit or 8 bit. Important: All registers must be accessed at their stated size. It is illegal to access a 16 bit register as 2 8 bit parts or 2 8 bit registers as a 16 bit register.

## **4.1 UART**

The UART is set up as a pair of 8-entry FIFO queues that feed the Tx and Rx circuitry.

| Name           | Size | Address | Description            |
|----------------|------|---------|------------------------|
| IO_UART_DATA   | 8    | 0xFF02  | UART data              |
| IO_UART_STATUS | 8    | 0xFF03  | UART status            |
| IO_UART_BAUD   | 16   | 0xFF04  | UART Baud rate divisor |

## 4.1.1 IO\_UART\_DATA

|   | Bits | Type | Description          |
|---|------|------|----------------------|
|   | 0-7  | W    | Data sent to UART Tx |
| Ì | 0-7  | R    | Data from UART Rx    |

#### 4.1.2 IO\_UART\_STATUS

| Bits | Type | Description        |
|------|------|--------------------|
| 0    | R    | Tx FIFO Space free |
| 1    | R    | Tx FIFO Empty      |
| 2    | R    | Rx Data ready      |
| 3    | R    | Rx FIFO Overrun    |
| 4-7  | X    | Reserved           |

## 4.1.3 IO\_UART\_BAUD

|      |     | Description       |
|------|-----|-------------------|
| 0-15 | R/W | Baud rate divisor |